home *** CD-ROM | disk | FTP | other *** search
/ EnigmA Amiga Run 1996 June / EnigmA AMIGA RUN 08 (1996)(G.R. Edizioni)(IT)[!][issue 1996-06][EARSAN CD VII].iso / earcd / utilsys / rss14gmd.lha / RSys_1.4gmd / C / SignalTrap.c < prev    next >
C/C++ Source or Header  |  1996-05-04  |  5KB  |  238 lines

  1. /*
  2.    ***************************************************************************
  3.    *
  4.    * Datei:
  5.    *      RSysSignalTrap.c
  6.    *
  7.    * Inhalt:
  8.    *
  9.    *      --- Globale Routinen ---
  10.    *
  11.    *    void HandleTrap ( int signal );
  12.    *    void InstallTrapHandlers ( void );
  13.    *    void RemoveTrapHandlers ( void );
  14.    *    void _traphandler ( long sig );
  15.    *
  16.    *      --- Lokale  Routinen ---
  17.    *
  18.    *    static void ( *_sigfuncs [_NUMSIG ]);
  19.    *    static void convert ( ULONG state , char *str );
  20.    *
  21.    * Bemerkungen:
  22.    *      Verwaltung eines Traphandlers, um RSys wirklich (?) absturzsicher
  23.    *      zu machen.
  24.    *
  25.    * Erstellungsdatum:
  26.    *      07-Jul-93     Rolf Böhme
  27.    *
  28.    * Änderungen:
  29.    *      07-Jul-93     Rolf Böhme        Erstellung
  30.    *
  31.    ***************************************************************************
  32.  */
  33.  
  34. #include "RSys.h"
  35. #include "protos.h"
  36.  
  37. /*
  38.    #define SIG_BEGINSSP                 0
  39.    #define SIG_BEGINPC                  1
  40.    #define SIG_BUSERROR                 2
  41.    #define SIG_ADDRERROR                3
  42.    #define SIG_ILLCOMM                  4
  43.    #define SIG_DIVZERO                  5
  44.    #define SIG_CHKERROR                 6
  45.    #define SIG_TRAPV                    7
  46.    #define SIG_VIOLPRIV                 8
  47.    #define SIG_TRACE                    9
  48.    #define SIG_LINEAEMUL                10
  49.    #define SIG_LINEFEMUL                11
  50.    #define SIG_NOTDEF                   12
  51.    #define SIG_COPROPROT                13
  52.    #define SIG_FORMATERR                14
  53.    #define SIG_INTRFAIL                 15
  54.  
  55.    #define SIG_MC_ILLCOND               48
  56.    #define SIG_MC_ILLRESULT     49
  57.    #define SIG_MC_DIVZERO               50
  58.    #define SIG_MC_UNDERFLOW     51
  59.    #define SIG_MC_OPERROR               52
  60.    #define SIG_MC_OVERFLOW      53
  61.    #define SIG_MC_NANSIG                54
  62.    #define SIG_MC_NOTDEF                55
  63.  */
  64.  
  65. static void (*_sigfuncs[_NUMSIG]) (int);
  66. int _trapexception (void);
  67. int _initfpu (void);
  68. void *_oldtrap, **_trapaddr;
  69.  
  70. static TASK *tp;
  71.  
  72. char *TrapCPUMsg[] =
  73. {
  74.   "Begin of SSP (ISP by 680x0)",
  75.   "Begin of PC",
  76.   "Bus error",
  77.   "Address error",
  78.   "Illegal or unknown instruction",
  79.   "Division by zero or arithmetic fault",
  80.   "CHK and CHK2 affected",
  81.   "TRAP or TRAPV reached",
  82.   "Priviledge violation",
  83.   "Trace error",
  84.   "Line A Emulator, illegal Opcode by MC68000",
  85.   "Line F Emulator, illegal Opcode by MC68000",
  86.   "Error is reserved",
  87.   "Coprocessor protocol error",
  88.   "Format error by MC680x0",
  89.   "Uninitialized Interrupt"
  90. };
  91.  
  92.  /*
  93.   * convert() konvertiert eine 32Bit-Zahl in einen
  94.   * "Binärstring", also in einen String, der nur aus '0'
  95.   * und '1' besteht
  96.   */
  97. static void
  98. convert (ULONG state, char *str)
  99. {
  100.   int bit;
  101.  
  102.   for (bit = 0; bit < 32; bit++)
  103.     str[31 - bit] = ((state & (1L << bit)) ? '1' : '0');
  104.  
  105.   str[32] = STRINGEND;
  106.  
  107. }
  108.  
  109.  /*
  110.   * HandleTrap() ist die Funktion, die im Falle eines
  111.   * System traps aktiviert wird. Diese Routine ruft
  112.   * entweder den System debugger ROMWack auf, oder
  113.   * verursacht einen Kaltstart des Rechners, falls dies
  114.   * gewünscht wird. Installiert wird diese Routine
  115.   * mit der ANSI-Routine signal() des
  116.   * Aztec-C-Compilers
  117.   */
  118. void
  119. HandleTrap (int signal)
  120. {
  121.   int action;
  122.   char exceptstr[33], signalstr[33], statestr[33];
  123.   UBYTE *fmt = (UBYTE *) "Error signal: %s (%ld)\n"
  124.   "Exception   : %s (0x%08lX)\n"
  125.   "Task signal : %s (0x%08lX)\n"
  126.   "CPU Status  : %s (0x%08lX)\n"
  127.   "Trap data   : %08lX\n"
  128.   "Except data : %08lX\n";
  129.   ULONG exc, sig, sta;
  130.   APTR UserStack;
  131.  
  132.   UserStack = SuperState ();
  133.  
  134.   exc = SetExcept (0L, 0L);
  135.   sig = SetSignal (0L, 0L);
  136.   sta = SetSR (0L, 0L);
  137.  
  138.   UserState (UserStack);
  139.  
  140.   convert (exc, exceptstr);
  141.   convert (sig, signalstr);
  142.   convert (sta, statestr);
  143.  
  144.   if (IntuitionBase != NULL)
  145.     action = MyEasyRequest (NULL,
  146.            (UBYTE *) NAME " *** Emergency trap failure message ***",
  147.      (UBYTE *) "Continue|Terminate RSys|Call system debugger|Reboot system",
  148.                 fmt, TrapCPUMsg[signal], signal,
  149.                 exceptstr, exc, signalstr, sig, statestr, sta,
  150.                 tp->tc_TrapData, tp->tc_ExceptData);
  151.   else
  152.     {
  153.       action = -1;
  154.       Printf (fmt, (ULONG) TrapCPUMsg[signal], signal,
  155.           exceptstr, exc, signalstr, sig, statestr, sta,
  156.           tp->tc_TrapData, tp->tc_ExceptData);
  157.     }
  158.  
  159.   switch (action)
  160.     {
  161.     case -1:
  162.       exit (0);
  163.       break;
  164.  
  165.     case 0:
  166.       ColdReboot ();
  167.       break;
  168.  
  169.     case 1:
  170.       break;
  171.  
  172.     case 2:
  173.       CloseAll ();
  174.       break;
  175.  
  176.     case 3:
  177.       Debug (0L);
  178.       CloseAll ();
  179.       break;
  180.     }
  181.  
  182. }
  183.  
  184.  /*
  185.   * InstallTrapHandlers() installiert einen Traphandler,
  186.   * der auf verschiedene Traps reagieren kann
  187.   */
  188. void
  189. InstallTrapHandlers (void)
  190. {
  191.   int i;
  192.  
  193.   tp = FindTask (NULL);
  194.  
  195.   Forbid ();
  196.  
  197.   _trapaddr = (void *) &tp->tc_TrapCode;
  198.   _oldtrap = tp->tc_TrapCode;
  199.  
  200.   tp->tc_TrapCode = (APTR) & _trapexception;
  201.  
  202.   if (SysBase->AttnFlags & AFF_68881)
  203.     _initfpu ();
  204.  
  205.   for (i = _FSTSIG; i <= _NUMSIG; i++)
  206.     _sigfuncs[i] = HandleTrap;
  207.  
  208.   Permit ();
  209.  
  210. }
  211.  
  212. void
  213. RemoveTrapHandlers (void)
  214. {
  215.   int i;
  216.  
  217.   Forbid ();
  218.  
  219.   if (_trapaddr)
  220.     *_trapaddr = _oldtrap;
  221.  
  222.   for (i = _FSTSIG; i <= _NUMSIG; i++)
  223.     _sigfuncs[i] = NULL;
  224.  
  225.   Permit ();
  226.  
  227. }
  228.  
  229. void
  230. _traphandler (long sig)
  231. {
  232.   void (*handler) (int);
  233.  
  234.   if ((handler = _sigfuncs[sig]) != NULL)
  235.     (*handler) (sig);
  236.  
  237. }
  238.